[id].vue 76 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267
  1. // 商品详情页面
  2. <template>
  3. <div>
  4. <StoreHeaderCat ref="headercat" @updateFllow="updateFllow" />
  5. <div class="sld_goods_detail">
  6. <!-- 内容区顶部固定 start -->
  7. <div class="contain_con" v-if="containCon">
  8. <div class="contain_content flex_row_center_center">
  9. <div class="contain_con_left flex_row_start_center">
  10. <span
  11. class="store_type"
  12. v-if="goodsDetail.data.storeInf.isOwnStore == 1"
  13. >{{ L["自营"] }}</span
  14. >
  15. <span class="store_title">{{
  16. goodsDetail.data.storeInf.storeName
  17. }}</span>
  18. </div>
  19. <div class="contain_con_right">
  20. <div class="goods_description_title flex_row_between_center">
  21. <div class="description_title_left flex_row_start_center">
  22. <span
  23. class="cursor_pointer"
  24. :class="{ description_active: goodsDesctionType == 'detail' }"
  25. @click="goodsDescType('detail')"
  26. >{{ L["商品详情"] }}</span
  27. >
  28. <span
  29. class="cursor_pointer"
  30. :class="{
  31. description_active: goodsDesctionType == 'evaluate',
  32. }"
  33. @click="goodsDescType('evaluate')"
  34. >{{ L["商品说明"] }}</span
  35. >
  36. <span
  37. class="cursor_pointer"
  38. :class="{
  39. description_active: goodsDesctionType == 'service',
  40. }"
  41. @click="goodsDescType('service')"
  42. >{{ L["商品服务"] }}</span
  43. >
  44. <span
  45. class="cursor_pointer"
  46. :class="{
  47. description_active: goodsDesctionType == 'salestore',
  48. }"
  49. @click="goodsDescType('salestore')"
  50. >{{ L["店铺热销"] }}</span
  51. >
  52. </div>
  53. </div>
  54. </div>
  55. </div>
  56. </div>
  57. <!-- 内容区顶部固定 end -->
  58. <div class="goods_detail_content self_background" v-if="goodsDetail.data">
  59. <!-- 商品所属分类 ,联系客服,关注店铺 start-->
  60. <div class="goods_about_con">
  61. <div class="goods_about flex_row_between_center">
  62. <div class="goods_classify">
  63. <span
  64. v-for="(item, index) in goodsDetail.data.categoryPath"
  65. :key="index"
  66. >
  67. <nuxt-link
  68. :to="`/goods/list/${calcProductName(item)}_v-${ goodsDetail.data.categoryIdPath[index] }_gid-${index + 1}${index > 0? '_pid-' + goodsDetail.data.categoryIdPath[index - 1]: '_pid-0'
  69. }`"
  70. target="_blank"
  71. >
  72. {{ item }}</nuxt-link
  73. >
  74. <i> > </i>
  75. </span>
  76. <span style="color: #333333">{{
  77. goodsDetail.data.goodsName
  78. }}</span>
  79. </div>
  80. <div class="goods_about_right flex_row_between_center">
  81. <router-link
  82. target="_blank"
  83. :to="'/store/'+ calcProductName(goodsDetail.data.storeInf.storeName) +'_'+ goodsDetail.data.storeInf.storeId"
  84. class="goods_about_store flex_row_between_center"
  85. v-if="goodsDetail.data.storeInf"
  86. >
  87. <span>{{ goodsDetail.data.storeInf.storeName }}</span>
  88. <span
  89. v-if="
  90. goodsDetail.data.storeInf &&
  91. goodsDetail.data.storeInf.isOwnStore == '1'
  92. "
  93. >{{ L["自营"] }}</span
  94. >
  95. </router-link>
  96. <div
  97. class="contact_service focus_store"
  98. @click="focusStore"
  99. v-if="goodsDetail.data.storeInf"
  100. >
  101. <img
  102. src="/goods/collection.png"
  103. alt=""
  104. v-show="goodsDetail.data.storeInf.isFollowStore == true"
  105. />
  106. <img
  107. src="/goods/no_collection.png"
  108. alt=""
  109. v-show="goodsDetail.data.storeInf.isFollowStore == false"
  110. />
  111. <span>
  112. {{
  113. goodsDetail.data.storeInf.isFollowStore
  114. ? L["取消关注"]
  115. : L["关注店铺"]
  116. }}
  117. </span>
  118. </div>
  119. </div>
  120. </div>
  121. </div>
  122. <!-- 商品所属分类 ,联系客服,关注店铺 end-->
  123. <!-- 商品主要信息 start -->
  124. <div class="main_con">
  125. <!-- 商品相关 start -->
  126. <div class="goods_des">
  127. <!-- 商品图片列表 start -->
  128. <div class="goods_des_left">
  129. <!-- 商品放大镜效果 start-->
  130. <div class="goods_main_picture">
  131. <div class="mask" v-if="[5, 6].includes(goodsDetail.data.state)">
  132. <div class="circle">{{ L['已下架'] }}</div>
  133. </div>
  134. <div
  135. class="preview-box"
  136. @mousemove="move($event)"
  137. @mouseleave="out($event)"
  138. @mouseenter="enter($event)"
  139. ref="previewBox"
  140. >
  141. <div
  142. class="imageBorder"
  143. :class="{
  144. default_image: true,
  145. skeleton_default_image: firstLoading,
  146. }"
  147. :style="{ backgroundImage: 'url(' + defaultImage + ')' }"
  148. >
  149. <video
  150. v-if="
  151. currentDefaultImage == -1 && goodsDetail.data.goodsVideo
  152. "
  153. controls
  154. playsinline="playsinline"
  155. class="imageBorder default_image"
  156. :poster="defaultImage"
  157. autoplay
  158. ref="video"
  159. >
  160. <source
  161. :src="goodsDetail.data.goodsVideo"
  162. type="video/mp4"
  163. />
  164. </video>
  165. </div>
  166. <div
  167. class="v_btn"
  168. v-if="
  169. currentDefaultImage != -1 && goodsDetail.data.goodsVideo
  170. "
  171. >
  172. <img src="/goods/playV.png" alt="" @click="playV" />
  173. </div>
  174. <!-- 遮罩 start-->
  175. <div class="mask" ref="maskBox" v-show="maskShow"></div>
  176. <!-- 遮罩 end -->
  177. <!-- 底部放大镜icon图标 start -->
  178. <div
  179. class="magnifier_icon flex_row_center_center"
  180. v-show="!maskShow"
  181. >
  182. <i class="iconfont icon-sousuo"></i>
  183. </div>
  184. <!-- 底部放大镜icon图标 end -->
  185. </div>
  186. <!-- 右侧的放大后的图片 start -->
  187. <div
  188. class="goods_picture_big"
  189. style="border: 1px solid #eee"
  190. ref="zoomBox"
  191. v-show="maskShow"
  192. >
  193. <div
  194. class="default_image_big"
  195. :style="{ backgroundImage: 'url(' + defaultImage + ')' }"
  196. ref="pictureBig"
  197. ></div>
  198. </div>
  199. <!-- 右侧的放大后的图片 end -->
  200. </div>
  201. <!-- 商品放大镜效果 end -->
  202. <!-- 商品图片列表 start -->
  203. <div
  204. :class="{
  205. goods_picture_con: true,
  206. flex_row_between_center: true,
  207. skeleton_goods_picture_con: firstLoading,
  208. }"
  209. v-if="
  210. goodsDetail.data.defaultProduct &&
  211. goodsDetail.data.defaultProduct.goodsPics &&
  212. goodsDetail.data.defaultProduct.goodsPics.length > 0
  213. "
  214. >
  215. <i
  216. class="iconfont icon-ziyuan2 left_arrow"
  217. :class="{ no_left_arrow: currentDefaultImage == 0 }"
  218. @click="switchDefaultImage('left')"
  219. ></i>
  220. <div class="show_box">
  221. <ul
  222. class="goods_picture_list flex_row_start_center"
  223. ref="goodsPictureList"
  224. >
  225. <li
  226. v-for="(goodsImgItem, goodsImgIndex) in goodsDetail.data
  227. .defaultProduct.goodsPics"
  228. :key="goodsImgIndex"
  229. class="goods_img"
  230. :class="{
  231. goods_img_active: currentDefaultImage == goodsImgIndex,
  232. }"
  233. @click="selectDefaultImage(goodsImgItem, goodsImgIndex)"
  234. @mouseover="
  235. selectDefaultImage(goodsImgItem, goodsImgIndex)
  236. "
  237. >
  238. <div
  239. class="goods_image"
  240. :style="{
  241. backgroundImage: 'url(' + goodsImgItem + ')',
  242. }"
  243. ></div>
  244. </li>
  245. </ul>
  246. </div>
  247. <i
  248. class="iconfont icon-ziyuan11 right_arrow"
  249. :class="{
  250. no_left_arrow:
  251. currentDefaultImage ==
  252. goodsDetail.data.defaultProduct.goodsPics.length - 1,
  253. }"
  254. @click="switchDefaultImage('right')"
  255. ></i>
  256. </div>
  257. <!-- 商品图片列表 end -->
  258. <!-- 商品分享和收藏 start -->
  259. <div
  260. class="collection_share_btn flex_row_start_start"
  261. v-if="goodsDetail.data.state == 3"
  262. >
  263. <div
  264. class="collection_btn flex_row_start_center cursor_pointer"
  265. @click="collectGoods"
  266. >
  267. <img
  268. src="/goods/collection.png"
  269. alt=""
  270. v-if="goodsDetail.data.followGoods"
  271. />
  272. <img src="/goods/collection1.png" alt="" v-else />
  273. <span>{{
  274. goodsDetail.data.followGoods ? L["已收藏"] : L["收藏"]
  275. }}</span>
  276. </div>
  277. <div class="share_btn">
  278. <AddThis publicId="ra-5ab34ca22008ed41" />
  279. </div>
  280. </div>
  281. <!-- 商品分享和收藏 end -->
  282. </div>
  283. <!-- 商品图片列表 end -->
  284. <!-- 商品详细信息 start -->
  285. <div class="m_item_inner">
  286. <div class="item_info">
  287. <div
  288. :class="{
  289. detaile_name: true,
  290. skeleton_detaile_name: firstLoading,
  291. }"
  292. >
  293. {{ goodsDetail.data.goodsName }}
  294. </div>
  295. <div
  296. :class="{ p_ad: true, skeleton_p_ad: firstLoading }"
  297. v-if="goodsDetail.data.goodsBrief"
  298. >
  299. {{ goodsDetail.data.goodsBrief }}
  300. </div>
  301. <!-- 商品未下架即正常商品 start -->
  302. <div v-if="goodsDetail.data.state == 3 || firstLoading">
  303. <div class="summary">
  304. <div
  305. class="sld_summary_item summary_goods clearfix"
  306. v-if="goodsDetail.data.defaultProduct"
  307. >
  308. <div class="sld_summary_goods_left">
  309. <div class="goods_price flex_row_start_center">
  310. <!-- 在售价 -->
  311. <div>
  312. <span class="price_title">{{ L["价格"] }}</span>
  313. <strong
  314. :class="{
  315. p_price: true,
  316. skeleton_p_price: firstLoading,
  317. }"
  318. >
  319. <span>
  320. {{
  321. goodsDetail.data.goodsMoney == null
  322. ? "面议"
  323. : goodsDetail.data.goodsMoney
  324. }}
  325. </span>
  326. </strong>
  327. </div>
  328. </div>
  329. </div>
  330. </div>
  331. </div>
  332. <div v-if="
  333. goodsDetail.data.goodsMinOrder != undefined &&
  334. goodsDetail.data.goodsMinOrder
  335. " class="goodsMinOrder">
  336. <span class="">{{ L["最低采购数量:"] }}</span>
  337. <strong>
  338. <span>
  339. {{
  340. goodsDetail.data.goodsMinOrder == null
  341. ? "--"
  342. : goodsDetail.data.goodsMinOrder
  343. }}
  344. </span>
  345. </strong>
  346. </div>
  347. <!-- 商品简介 start -->
  348. <div class="summary-info">
  349. <div
  350. v-if="
  351. goodsDetail.data.goodsSummary != undefined &&
  352. goodsDetail.data.goodsSummary
  353. "
  354. class="summary_html"
  355. :style="goodsDetail.data.goodsSummaryBg"
  356. >
  357. <div
  358. class="summary_htmls"
  359. v-html="goodsDetail.data.goodsSummary"
  360. ></div>
  361. </div>
  362. </div>
  363. <!-- 规格 start -->
  364. <div
  365. class="goods_spec"
  366. v-if="
  367. goodsDetail.data.specs &&
  368. goodsDetail.data.specs.length > 0
  369. "
  370. >
  371. <div
  372. class="goods_spec_pre flex_row_start_start"
  373. v-for="(specItem, specIndex) in goodsDetail.data.specs"
  374. :key="specIndex"
  375. >
  376. <div
  377. class="goods_spec_pre_title"
  378. :title="specItem.specName"
  379. >
  380. {{ specItem.specName }}
  381. </div>
  382. <div class="goods_spec_pre_list flex_row_start_center">
  383. <!-- checkState : 1-选中,2-可选,3-禁用 -->
  384. <!-- 禁止选择 -->
  385. <div
  386. class="specval_pre cursor_pointer"
  387. :class="{
  388. specval_pre_disabled: sepcValItem.checkState == '3',
  389. }"
  390. v-for="(
  391. sepcValItem, specValIndex
  392. ) in specItem.specValueList"
  393. :key="specValIndex"
  394. v-show="sepcValItem.checkState == '3'"
  395. >
  396. <div
  397. class="specval_pre_image"
  398. :style="{
  399. backgroundImage: 'url(' + sepcValItem.image + ')',
  400. }"
  401. :title="sepcValItem.specValue"
  402. v-if="sepcValItem.image"
  403. ></div>
  404. <span class="specval_pre_text" v-else>{{
  405. sepcValItem.specValue
  406. }}</span>
  407. </div>
  408. <!-- 可选择 -->
  409. <div
  410. class="specval_pre cursor_pointer"
  411. :class="{
  412. specval_pre_active: sepcValItem.checkState == '1',
  413. }"
  414. v-for="(
  415. sepcValItem, specValIndex
  416. ) in specItem.specValueList"
  417. :key="specValIndex"
  418. @click="
  419. selectSpecVal(
  420. 'choice',
  421. specItem.specId,
  422. sepcValItem.specValueId
  423. )
  424. "
  425. v-show="sepcValItem.checkState != '3'"
  426. >
  427. <div
  428. class="goods_image"
  429. :style="{
  430. backgroundImage: 'url(' + goodsImgItem + ')',
  431. }"
  432. ></div>
  433. <div
  434. class="specval_pre_image"
  435. :style="{
  436. backgroundImage: 'url(' + sepcValItem.image + ')',
  437. }"
  438. :title="sepcValItem.specValue"
  439. v-if="sepcValItem.image"
  440. ></div>
  441. <span class="specval_pre_text" v-else>{{
  442. sepcValItem.specValue
  443. }}</span>
  444. <img
  445. src="/goods/check_mark.png"
  446. alt=""
  447. class="check_mark"
  448. v-if="sepcValItem.checkState == '1'"
  449. />
  450. </div>
  451. </div>
  452. </div>
  453. </div>
  454. <!-- 规格 end -->
  455. </div>
  456. <!-- 商品未下架即正常商品 end -->
  457. <template v-if="!firstLoading">
  458. <!-- 立即购买 加入购物车 收藏 ,分享 start-->
  459. <!-- 商品已下架 start -->
  460. <div
  461. class="options_btn"
  462. v-if="goodsDetail.data.state && goodsDetail.data.state != 3"
  463. >
  464. <p class="option_desc">{{L['商品已下架,欢迎挑选其他商品~']}}</p>
  465. <div class="goods_off_shelves">{{ L["商品已下架"] }}</div>
  466. <!-- 商品下架时的推荐商品 start -->
  467. <div class="recoOffShop">
  468. <div
  469. class="reCon"
  470. v-for="(
  471. {
  472. goodsImage,
  473. goodsName,
  474. goodsPrice,
  475. defaultProductId,
  476. },
  477. index
  478. ) in recomOffShop.data"
  479. :key="index"
  480. >
  481. <div class="reComImg">
  482. <router-link
  483. :to="'/goods/detail/'+ calcProductName(goodsName) +'_'+ defaultProductId"
  484. target="_blank"
  485. >
  486. <img :src="goodsImage" alt="" />
  487. </router-link>
  488. </div>
  489. <router-link
  490. class="recomName"
  491. :to="'/goods/detail/'+ calcProductName(goodsName) +'_'+ defaultProductId"
  492. target="_blank"
  493. >
  494. {{ goodsName }}
  495. </router-link>
  496. <span class="recomPrice">{{ goodsMoney }}</span>
  497. </div>
  498. </div>
  499. <!-- 商品下架时的推荐商品 end -->
  500. </div>
  501. <!-- 商品已下架 end -->
  502. <!-- 普通(活动)正常商品 start -->
  503. <div class="options_btn flex_row_start_center" v-else>
  504. <div
  505. class="goods_code"
  506. id="qrcodeAct"
  507. v-show="isShowQr"
  508. ></div>
  509. <div v-if="goodsDetail.data.state === 3" class="buy_now flex_row_center_center mt-20" @click="goBuy">
  510. {{ L["发送询盘"] }}
  511. </div>
  512. </div>
  513. <!-- 普通(活动)正常商品 end -->
  514. <!-- 立即购买 加入购物车 收藏 ,分享 end-->
  515. </template>
  516. </div>
  517. </div>
  518. <!-- 商品详细信息 end -->
  519. <!-- 相关推荐 start -->
  520. <div class="more_goods" v-if="goodsDetail.data.state == 3">
  521. <div class="more_goods_title">{{ L["看了又看"] }}</div>
  522. <div class="more_goods_list flex_column_center_center">
  523. <template v-if="firstLoading">
  524. <div
  525. class="more_goods_pre"
  526. v-for="(recommendItem, recommendIndex) in [
  527. { a: 1 },
  528. { b: 2 },
  529. { c: 3 },
  530. ]"
  531. :key="recommendIndex"
  532. >
  533. <router-link
  534. target="_blank"
  535. :to="'/goods/detail/'+ calcProductName(recommendItem.goodsName) +'_'+ recommendItem.defaultProductId"
  536. >
  537. <div
  538. :class="{
  539. more_goods_pre_img: true,
  540. skeleton_more_goods_pre_img: firstLoading,
  541. }"
  542. ></div>
  543. <p
  544. :class="{
  545. skeleton_more_goods_pre_goods_name: firstLoading,
  546. }"
  547. >
  548. {{ recommendItem.goodsName }}
  549. </p>
  550. <p>
  551. <span
  552. :class="{
  553. skeleton_more_goods_pre_goods_price: firstLoading,
  554. }"
  555. ></span>
  556. </p>
  557. </router-link>
  558. </div>
  559. </template>
  560. <template v-else>
  561. <div
  562. class="more_goods_pre"
  563. v-for="(
  564. recommendItem, recommendIndex
  565. ) in recommendeList.data"
  566. :key="recommendIndex"
  567. v-show="recommendIndex < 2"
  568. >
  569. <router-link
  570. target="_blank"
  571. :to="'/goods/detail/'+ calcProductName(recommendItem.goodsName) +'_'+ recommendItem.defaultProductId"
  572. >
  573. <div class="more_goods_pre_img flex_row_center_center">
  574. <img
  575. :src="recommendItem.goodsImage"
  576. :title="recommendItem.goodsName"
  577. />
  578. </div>
  579. <p>{{ recommendItem.goodsName }}</p>
  580. <p>{{ recommendItem.goodsMoney }}</p>
  581. </router-link>
  582. </div>
  583. </template>
  584. </div>
  585. </div>
  586. <!-- 相关推荐 end -->
  587. </div>
  588. <!-- 商品相关 end -->
  589. <!-- 店铺,及各种信息的切换 start -->
  590. <div
  591. class="container flex_row_start_start"
  592. ref="container"
  593. id="container"
  594. >
  595. <div class="left">
  596. <div class="store_info" v-if="goodsDetail.data.storeInf">
  597. <div class="store_info_title flex_row_start_center">
  598. <span
  599. class="store_type"
  600. v-if="goodsDetail.data.storeInf.isOwnStore == 1"
  601. >{{ L["自营"] }}</span
  602. >
  603. <router-link
  604. target="_blank"
  605. :to="'/store/'+ calcProductName(goodsDetail.data.storeInf.storeName) +'_'+ goodsDetail.data.storeInf.storeId"
  606. >
  607. <span class="store_title" @click="goStore()">{{
  608. goodsDetail.data.storeInf.storeName
  609. }}</span>
  610. </router-link>
  611. </div>
  612. <div class="store_des">
  613. <div class="store_des_pre pre_service">
  614. <span>{{ L["客服电话"] }}:</span>
  615. <span>{{ goodsDetail.data.storeInf.servicePhone }}</span>
  616. </div>
  617. <div class="store_des_pre pre_service">
  618. <span>{{ L["主营商品"] }}:</span>
  619. <span>{{ goodsDetail.data.storeInf.mainBusiness }}</span>
  620. </div>
  621. </div>
  622. <div class="store_btn flex_row_center_center">
  623. <div class="store_btn_pre flex_row_center_center">
  624. <img src="/goods/store.png" alt="" class="btn_pre_img" />
  625. <router-link
  626. target="_blank"
  627. :to="'/store/'+ calcProductName(goodsDetail.data.storeInf.storeName) +'_'+ goodsDetail.data.storeInf.storeId"
  628. class="go_store_btn"
  629. >
  630. {{ L["进入店铺"] }}
  631. </router-link>
  632. </div>
  633. <div
  634. class="store_btn_pre flex_row_center_center"
  635. @click="focusStore"
  636. >
  637. <img
  638. src="/goods/collection.png"
  639. alt=""
  640. v-if="goodsDetail.data.storeInf.isFollowStore"
  641. />
  642. <img src="/goods/no_collection.png" alt="" v-else />
  643. <span>{{
  644. goodsDetail.data.storeInf.isFollowStore
  645. ? L["取消关注"]
  646. : L["关注店铺"]
  647. }}</span>
  648. </div>
  649. </div>
  650. </div>
  651. <!-- 店铺推荐及热门收藏 end -->
  652. </div>
  653. <!-- 商品详情 评价 商品服务 店铺热销 start-->
  654. <div class="goods_description">
  655. <div class="goods_description_title flex_row_between_center">
  656. <div class="description_title_left flex_row_start_center">
  657. <span
  658. class="cursor_pointer"
  659. :class="{
  660. description_active: goodsDesctionType == 'detail',
  661. }"
  662. @click="goodsDescType('detail')"
  663. >{{ L["商品详情"] }}</span
  664. >
  665. <span
  666. class="cursor_pointer"
  667. v-if="
  668. goodsDetail.data.goodsAnnexList &&
  669. goodsDetail.data.goodsAnnexList.length > 0
  670. "
  671. :class="{
  672. description_active: goodsDesctionType == 'evaluate',
  673. }"
  674. @click="goodsDescType('evaluate')"
  675. >{{ L["商品说明"] }}</span
  676. >
  677. <span
  678. class="cursor_pointer"
  679. v-if="
  680. goodsDetail.data.serviceLabels &&
  681. goodsDetail.data.serviceLabels.length > 0
  682. "
  683. :class="{
  684. description_active: goodsDesctionType == 'service',
  685. }"
  686. @click="goodsDescType('service')"
  687. >{{ L["商品服务"] }}</span
  688. >
  689. <span
  690. class="cursor_pointer"
  691. :class="{
  692. description_active: goodsDesctionType == 'salestore',
  693. }"
  694. @click="goodsDescType('salestore')"
  695. >{{ L["店铺推荐"] }}</span
  696. >
  697. </div>
  698. </div>
  699. <div class="goods_description_con">
  700. <!-- 商品详情,规格参数,及详情富文本 start-->
  701. <div class="goods_des_con" v-if="goodsDesctionType == 'detail'">
  702. <div
  703. v-if="
  704. goodsDetail.data.brandName ||
  705. (goodsDetail.data.goodsParameterList &&
  706. goodsDetail.data.goodsParameterList.length > 0) ||
  707. goodsDetail.data.goodsDetails
  708. "
  709. >
  710. <div class="brand" v-if="goodsDetail.data.brandName">
  711. <span>{{ L["品牌"] }}: </span>
  712. <span>{{ goodsDetail.data.brandName }}</span>
  713. </div>
  714. <div
  715. v-if="
  716. goodsDetail.data.goodsParameterList &&
  717. goodsDetail.data.goodsParameterList.length > 0
  718. "
  719. >
  720. <div
  721. class="goods_parameter_list"
  722. :class="{ goods_paramter_more: !parameterShow }"
  723. >
  724. <div
  725. class="goods_parameter_pre"
  726. v-for="(parameterItem, paramterIndex) in goodsDetail
  727. .data.goodsParameterList"
  728. :key="paramterIndex"
  729. >
  730. <span>{{ parameterItem.parameterName }}: </span>
  731. <span>{{ parameterItem.parameterValue }}</span>
  732. </div>
  733. </div>
  734. <div
  735. class="collapse_unfold flex_row_center_center cursor_pointer"
  736. v-if="goodsDetail.data.goodsParameterList.length > 16"
  737. @click="openParameter"
  738. >
  739. <span>{{
  740. !parameterShow ? L["查看全部"] : L["收起全部"]
  741. }}</span>
  742. <i
  743. class="iconfont icon-ziyuan11-copy"
  744. v-if="!parameterShow"
  745. ></i>
  746. <i class="iconfont icon-ziyuan11-copy-copy" v-else></i>
  747. </div>
  748. </div>
  749. <div
  750. v-if="
  751. goodsDetail.data.topTemplateContent != undefined &&
  752. goodsDetail.data.topTemplateContent
  753. "
  754. class="goods_html"
  755. :style="goodsDetail.data.topTemplateContentBg"
  756. >
  757. <div
  758. class="goods_htmls"
  759. v-html="goodsDetail.data.topTemplateContent"
  760. ></div>
  761. </div>
  762. <div
  763. v-if="
  764. goodsDetail.data.goodsDetails != undefined &&
  765. goodsDetail.data.goodsDetails
  766. "
  767. class="goods_html"
  768. :style="goodsDetail.data.goodsDetailsBg"
  769. >
  770. <div
  771. class="goods_htmls"
  772. v-html="goodsDetail.data.goodsDetails"
  773. ></div>
  774. </div>
  775. <div
  776. v-if="
  777. goodsDetail.data.bottomTemplateContent != undefined &&
  778. goodsDetail.data.bottomTemplateContent
  779. "
  780. class="goods_html"
  781. :style="goodsDetail.data.bottomTemplateContentBg"
  782. >
  783. <div
  784. class="goods_htmls"
  785. v-html="goodsDetail.data.bottomTemplateContent"
  786. ></div>
  787. </div>
  788. </div>
  789. <div v-else>
  790. <SldCommonEmpty
  791. :tip="L['该商品暂无详情~']"
  792. totalWidth="934px"
  793. />
  794. </div>
  795. </div>
  796. <!-- 商品详情,规格参数,及详情富文本 end-->
  797. <!-- 商品评价 start -->
  798. <div
  799. class="goods_comments"
  800. v-if="goodsDesctionType == 'evaluate'"
  801. >
  802. <!--说明书下载begin-->
  803. <div
  804. class="download_warp"
  805. v-if="
  806. goodsDetail.data.goodsAnnexList &&
  807. goodsDetail.data.goodsAnnexList.length > 0
  808. "
  809. >
  810. <span style="margin: 10px; font-size: 14px"
  811. >{{ L["说明书下载"] }}:
  812. </span>
  813. <br />
  814. <p
  815. style="margin: 10px; font-size: 14px"
  816. v-for="(item, index) in goodsDetail.data.goodsAnnexList"
  817. :key="index"
  818. >
  819. <a
  820. style="text-decoration: underline"
  821. href="javascript:;"
  822. @click="downloadAdd(item)"
  823. >{{ item.annexName }}</a
  824. >
  825. </p>
  826. </div>
  827. <div v-else>
  828. {{ L["暂无说明"] }}
  829. </div>
  830. <!--说明书下载end-->
  831. </div>
  832. <!-- 商品评价 end -->
  833. <!-- 商品服务 start -->
  834. <div
  835. class="goods_server_list"
  836. v-if="goodsDesctionType == 'service'"
  837. >
  838. <div
  839. v-if="
  840. goodsDetail.data.serviceLabels &&
  841. goodsDetail.data.serviceLabels.length > 0
  842. "
  843. >
  844. <div
  845. class="goods_server_pre"
  846. v-for="(serverItem, serverIndex) in goodsDetail.data
  847. .serviceLabels"
  848. :key="serverIndex"
  849. >
  850. <div class="server_pre_top flex_row_start_center">
  851. <span class="server_pre_tips"></span>
  852. <span class="server_pre_name">{{
  853. serverItem.labelName
  854. }}</span>
  855. </div>
  856. <div class="server_pre_content">
  857. {{ serverItem.description }}
  858. </div>
  859. </div>
  860. </div>
  861. <div v-else>
  862. <SldCommonEmpty
  863. :tip="L['暂无商品服务~']"
  864. totalWidth="934px"
  865. />
  866. </div>
  867. </div>
  868. <!-- 商品服务 end -->
  869. <!-- 店铺推荐 start -->
  870. <div
  871. class="store_hot_sales"
  872. v-if="goodsDesctionType == 'salestore'"
  873. >
  874. <div
  875. v-if="
  876. recommendedList.data && recommendedList.data.length > 0
  877. "
  878. >
  879. <div class="store_hot_sales_list">
  880. <div
  881. class="goods_pre flex_column_between_start"
  882. v-for="(
  883. recommendItem, recommendIndex
  884. ) in recommendedList.data"
  885. :key="recommendIndex"
  886. >
  887. <router-link
  888. target="_blank"
  889. :to="'/goods/detail/'+ calcProductName(recommendItem.goodsName) +'_'+ recommendItem.defaultProductId"
  890. >
  891. <div class="flex_column_start_start">
  892. <div
  893. class="goods_pre_img"
  894. :style="{
  895. backgroundImage:
  896. 'url(' + recommendItem.goodsImage + ')',
  897. }"
  898. ></div>
  899. <div class="goods_name">
  900. {{ recommendItem.goodsName }}
  901. </div>
  902. </div>
  903. <div class="goods_price">
  904. <div class="selling_price">
  905. {{ recommendItem.goodsMoney }}
  906. </div>
  907. </div>
  908. </router-link>
  909. </div>
  910. </div>
  911. <div
  912. class="flex_row_end_center sld_pagination sld_page_bottom"
  913. v-if="
  914. recommendeData.data.pagination &&
  915. recommendeData.data.pagination.total
  916. "
  917. >
  918. <el-pagination
  919. @current-change="handleCurrentChangeSales"
  920. v-model:currentPage="salesCurrent"
  921. :page-size="salesPageSize"
  922. layout="prev, pager, next, jumper"
  923. :total="recommendeData.data.pagination.total"
  924. :hide-on-single-page="true"
  925. >
  926. </el-pagination>
  927. </div>
  928. </div>
  929. <div
  930. v-if="
  931. recommendedList.data && recommendedList.data.length == 0
  932. "
  933. class="flex_column_center_center empty_data"
  934. >
  935. <SldCommonEmpty
  936. :tip="L['暂无相关商品~']"
  937. totalWidth="934px"
  938. />
  939. </div>
  940. </div>
  941. <!-- 店铺推荐 end -->
  942. </div>
  943. </div>
  944. <!-- 商品详情 评价 商品服务 店铺热销 end-->
  945. </div>
  946. <!-- 店铺,及各种信息的切换 end -->
  947. </div>
  948. <!-- 商品主要信息 end -->
  949. </div>
  950. <SldLoginModal
  951. v-if="loginModalVisibleFlag"
  952. @closeLoingModal="closeLoingModal"
  953. @refreshInfo="refreshInfo"
  954. />
  955. <EnquiryModal
  956. v-if="enquiryVis"
  957. :itemType="'GOODS'"
  958. :itemId="productId"
  959. @closeLoingModal="closeEnquiryModal"
  960. />
  961. </div>
  962. </div>
  963. </template>
  964. <script setup>
  965. import addrData from "@/assets/area.json";
  966. import { ElMessage, ElRate, ElDialog, ElPagination } from "element-plus";
  967. import { qrcanvas } from "qrcanvas";
  968. // import { lang_zn } from "@/assets/language/zh";
  969. import { getCurLanguage } from '@/composables/common.js';
  970. import { goodsInfo, useUserInfo } from "@/store/user.js";
  971. import { useFiltersStore } from "@/store/filter.js";
  972. const filtersStore = useFiltersStore();
  973. const goodsInfox = goodsInfo();
  974. const configInfo = useUserInfo();
  975. // const L = lang_zn;
  976. const L = getCurLanguage();
  977. const firstLoading = ref(true); //是否第一次加载
  978. const router = useRouter();
  979. const route = useRoute();
  980. const store = ref();
  981. const imgVisible = ref(false);
  982. const imgSource = ref("");
  983. const imgIndex = ref(-1);
  984. const vid = ref(0); //店铺id
  985. const proxy = getCurrentInstance();
  986. const goodsDetail = reactive({ data: {} }); //商品详情数据
  987. const isChoice = ref("default"); //是默认选中的,还是点击选择规格之后的 default:默认 choice:选择
  988. const productId = ref(""); //货品id
  989. const fullDisList = reactive({ data: [] }); //满优惠促销列表
  990. const couponList = reactive({ data: [] }); //获取店铺优惠券列表
  991. const pictureBig = ref(null); //大图的信息
  992. const maskBox = ref(null); //遮罩盒子的信息
  993. const maskShow = ref(false); //遮罩是否显示
  994. const previewBox = ref(null); // 左侧主图元素信息
  995. const zoomBox = ref(null); //左侧主图的父元素的信息
  996. const defaultImage = ref(""); //默认主图路径
  997. const currentDefaultImage = ref("0"); //默认主图显示第一张
  998. const currentSpecNum = ref(1); //商品编辑数量,默认数量为1
  999. const recommendeList = reactive({ data: [] }); //看了又看商品
  1000. const storePopularList = reactive({ data: [] }); //店铺推荐及热门收藏
  1001. const storePopularType = ref("collection"); //店铺推荐及热门收藏,默认显示店铺推荐
  1002. const goodsDesctionType = ref("detail"); //商品详情,评价,商品服务,店铺热销,默认显示商品详情
  1003. const goodsCommentsInfo = reactive({ data: {} }); //商品评价信息
  1004. const evaluationType = ref(""); //商品评价类型,默认显示全部
  1005. const recommendedList = reactive({ data: [] }); //店铺推荐列表
  1006. const evaluationCurrent = ref(1); //评价列表默认第一页
  1007. const evaluationPageSize = ref(5); //评价列表默认一页显示5条数据
  1008. const salesCurrent = ref(1); //店铺热销列表默认一页
  1009. const salesPageSize = ref(20); //店铺热销列表默认一页显示20条数据
  1010. const recommendeData = reactive({ data: {} }); //店铺热销推荐数据
  1011. const couponModel = ref(false); //优惠券弹框是否显示
  1012. const cartListData = reactive({ data: goodsInfox.cartListData }); //获取vux的store中的购物车数据
  1013. const loginModalVisibleFlag = ref(false); //登录弹框是否显示,默认不显示
  1014. const enquiryVis = ref(false); //登录弹框是否显示,默认不显示
  1015. const container = ref(null); // 商品详情页底部内容区
  1016. const containerTop = ref(0); //商品详情页底部内容区的top值
  1017. const containCon = ref(false); //固定内容区头部
  1018. const fullDiscountModel = ref(false); //满优惠弹框是否显示
  1019. const goodsPictureList = ref(null); //商品图片列表
  1020. const curCouponPage = ref(1); //当前为第一页优惠券
  1021. const couponPageSize = ref(6); //优惠券默认一页显示6条数据
  1022. const wxShareCode = ref(false); //微信分享二维码是否显示
  1023. const parameterShow = ref(false); //规格参数查看是否查看全部,默认为否
  1024. const score = ref(0); //好评率
  1025. const colors = ref(["#E2231A", "#E2231A", "#E2231A"]); //星星颜色
  1026. const curAddr = ref(-1);
  1027. const curAddrName = ref("");
  1028. const addrIdx = ref(0);
  1029. const otherAddrIdx = ref(0);
  1030. const othTopIdx = reactive({
  1031. 0: 0,
  1032. 1: 0,
  1033. 2: 0,
  1034. });
  1035. const otherTree = ref([
  1036. addrData[othTopIdx["0"]],
  1037. addrData[othTopIdx["0"]].children[othTopIdx["1"]],
  1038. ]);
  1039. const othAddrDe = ref(addrData);
  1040. const addrDialogVisible = ref(false);
  1041. const logFlag = ref(configInfo.loginFlag);
  1042. // 促销活动信息
  1043. const preSellInfo = reactive({ data: {} });
  1044. const pinInfo = reactive({ data: {} });
  1045. const seckillInfo = reactive({ data: {} });
  1046. const ladderInfo = reactive({ data: {} });
  1047. const address_list = reactive({ data: [] });
  1048. const isShowQr = ref(false);
  1049. const secInt = ref("");
  1050. const time = reactive({
  1051. day: "00",
  1052. hours: "00",
  1053. minutes: "00",
  1054. seconds: "00",
  1055. });
  1056. const judgeStock = computed(() => {
  1057. return (
  1058. goodsDetail.data.defaultProduct.productStock == 0 ||
  1059. (JSON.stringify(preSellInfo.data) != "{}" &&
  1060. preSellInfo.data.presellStock == 0) ||
  1061. (JSON.stringify(pinInfo.data) != "{}" && pinInfo.data.spellStock == 0) ||
  1062. (JSON.stringify(seckillInfo.data) != "{}" &&
  1063. seckillInfo.data.seckillStock == 0)
  1064. );
  1065. });
  1066. // 促销活动信息end
  1067. const scrollHandle = async (e) => {
  1068. if (process.client) {
  1069. let elementScrollTop = e.srcElement.scrollingElement.scrollTop; //获取页面滚动高度
  1070. if (
  1071. document.getElementById("container") &&
  1072. elementScrollTop > document.getElementById("container").offsetTop
  1073. ) {
  1074. containCon.value = true;
  1075. await proxy.$nextTick();
  1076. } else {
  1077. containCon.value = false;
  1078. }
  1079. }
  1080. };
  1081. // 点击播放视频
  1082. const playV = () => {
  1083. currentDefaultImage.value = -1;
  1084. defaultImage.value = "";
  1085. maskShow.value = false;
  1086. videoEnd();
  1087. };
  1088. //获取商品详情数据
  1089. const getInitDataStatic = async (proId) => {
  1090. let params = {
  1091. productId: proId,
  1092. };
  1093. const { data: value, pending: pending } = await useFetchRaw(
  1094. apiUrl + "v3/goods/front/goods/details",
  1095. { params: params }
  1096. );
  1097. const res = value._rawValue;
  1098. if (res.state == 200) {
  1099. let staticData = [
  1100. "brandId",
  1101. "brandName",
  1102. "categoryPath",
  1103. "categoryIdPath",
  1104. "goodsBrief",
  1105. "goodsDetails",
  1106. "goodsSummary",
  1107. "goodsBrief",
  1108. "goodsId",
  1109. "goodsName",
  1110. "goodsParameterList",
  1111. "goodsVideo",
  1112. "topTemplateContent",
  1113. "bottomTemplateContent",
  1114. "goodsAnnexList",
  1115. "serviceLabels",
  1116. "goodsMoney",
  1117. "goodsMinOrder",
  1118. "goodsParameterList",
  1119. ];
  1120. staticData.forEach((item) => {
  1121. if (item == "categoryPath") {
  1122. goodsDetail.data.categoryPath = res.data.categoryPath.split("->");
  1123. } else if (item == "categoryIdPath") {
  1124. goodsDetail.data.categoryIdPath = [
  1125. res.data.categoryId1,
  1126. res.data.categoryId2,
  1127. res.data.categoryId3,
  1128. ];
  1129. } else {
  1130. goodsDetail.data[item] = res.data[item];
  1131. }
  1132. });
  1133. if (goodsDetail.data.goodsSummary) {
  1134. goodsDetail.data.goodsSummary = quillEscapeToHtml(
  1135. goodsDetail.data.goodsSummary
  1136. );
  1137. //处理背景样式
  1138. if (
  1139. goodsDetail.data.goodsSummary.indexOf(
  1140. '<p style="display:none;" data-background="'
  1141. ) != -1
  1142. ) {
  1143. let bg = goodsDetail.data.goodsSummary
  1144. .split('<p style="display:none;" data-background="')[1]
  1145. .split('">')[0];
  1146. goodsDetail.data.goodsSummaryBg = bg;
  1147. }
  1148. }
  1149. if (goodsDetail.data.topTemplateContent) {
  1150. goodsDetail.data.topTemplateContent = quillEscapeToHtml(
  1151. goodsDetail.data.topTemplateContent
  1152. );
  1153. //处理背景样式
  1154. if (
  1155. goodsDetail.data.topTemplateContent.indexOf(
  1156. '<p style="display:none;" data-background="'
  1157. ) != -1
  1158. ) {
  1159. let bg = goodsDetail.data.topTemplateContent
  1160. .split('<p style="display:none;" data-background="')[1]
  1161. .split('">')[0];
  1162. goodsDetail.data.topTemplateContentBg = bg;
  1163. }
  1164. }
  1165. if (goodsDetail.data.goodsDetails) {
  1166. goodsDetail.data.goodsDetails = quillEscapeToHtml(
  1167. goodsDetail.data.goodsDetails
  1168. );
  1169. //处理背景样式
  1170. if (
  1171. goodsDetail.data.goodsDetails.indexOf(
  1172. '<p style="display:none;" data-background="'
  1173. ) != -1
  1174. ) {
  1175. let bg = goodsDetail.data.goodsDetails
  1176. .split('<p style="display:none;" data-background="')[1]
  1177. .split('">')[0];
  1178. goodsDetail.data.goodsDetailsBg = bg;
  1179. }
  1180. }
  1181. if (goodsDetail.data.bottomTemplateContent) {
  1182. goodsDetail.data.bottomTemplateContent = quillEscapeToHtml(
  1183. goodsDetail.data.bottomTemplateContent
  1184. );
  1185. //处理背景样式
  1186. if (
  1187. goodsDetail.data.bottomTemplateContent.indexOf(
  1188. '<p style="display:none;" data-background="'
  1189. ) != -1
  1190. ) {
  1191. let bg = goodsDetail.data.bottomTemplateContent
  1192. .split('<p style="display:none;" data-background="')[1]
  1193. .split('">')[0];
  1194. goodsDetail.data.bottomTemplateContentBg = bg;
  1195. }
  1196. }
  1197. currentDefaultImage.value = 0;
  1198. vid.value = res.data.storeInf.storeId;
  1199. // setTimeout(() => {
  1200. // sldStatEvent({ behaviorType: 'gpv', goodsId: goodsDetail.data.goodsId, storeId: vid.value });
  1201. // }, 3000)
  1202. } else {
  1203. ElMessage.error(res.msg);
  1204. }
  1205. if (!pending._rawValue) {
  1206. getInitDataDynamic(productId.value);
  1207. getRecommend();
  1208. getStorePopular();
  1209. getEvaluation();
  1210. addLog();
  1211. }
  1212. };
  1213. getInitDataStatic(calcProductId(route.path));
  1214. onMounted(() => {
  1215. setTimeout(() => {
  1216. sldStatEvent({ behaviorType: 'gpv', goodsId: goodsDetail.data.goodsId, storeId: vid.value,pageUrl: defaultUrl + router.currentRoute.value.path, referrerPageUrl: apiUrl });
  1217. }, 3000)
  1218. });
  1219. productId.value = calcProductId(route.path);
  1220. if (!logFlag.value) {
  1221. addrIdx.value = 1;
  1222. }
  1223. const getInitDataDynamic = async (proId, updateType) => {
  1224. let params = {
  1225. productId: proId,
  1226. };
  1227. const { data: value, pending: pending } = await useFetchRaw(
  1228. apiUrl + "v3/goods/front/goods/details2",
  1229. { params: params,
  1230. headers:{Authorization:'Bearer ' + filtersStore.getToken}
  1231. }
  1232. );
  1233. const res = value._rawValue;
  1234. if (res.state == 200) {
  1235. useHead({
  1236. title: res.data.seoInfo.seoTitle || 'Goods Detail',
  1237. meta: [
  1238. {
  1239. name: "description",
  1240. content: res.data.seoInfo.seoDesc,
  1241. },
  1242. {
  1243. name: "keywords",
  1244. content: res.data.seoInfo.seoKeywords,
  1245. },
  1246. ],
  1247. });
  1248. defaultImage.value = res.data.defaultProduct.goodsPics[0];
  1249. let dynamicData = [
  1250. "defaultProduct",
  1251. "deliverInfo",
  1252. "effectSpecValueIds",
  1253. "followGoods",
  1254. "specs",
  1255. "storeInf",
  1256. "sales",
  1257. "state",
  1258. "shareLink",
  1259. "shareImage",
  1260. "goodsMinOrder",
  1261. "isVirtualGoods",
  1262. ];
  1263. dynamicData.forEach((item) => {
  1264. goodsDetail.data[item] = res.data[item];
  1265. });
  1266. if (goodsDetail.data.state != 3) {
  1267. getRecom();
  1268. }
  1269. if (goodsDetail.data.defaultProduct.promotionType == 103) {
  1270. getPreSell(res.data.defaultProduct.productId);
  1271. } else if (goodsDetail.data.defaultProduct.promotionType == 102) {
  1272. getPin();
  1273. } else if (goodsDetail.data.defaultProduct.promotionType == 104) {
  1274. getSeckill();
  1275. } else if (goodsDetail.data.defaultProduct.promotionType == 105) {
  1276. getLadder();
  1277. } else {
  1278. preSellInfo.data = {};
  1279. pinInfo.data = {};
  1280. seckillInfo.data = {};
  1281. ladderInfo.data = {};
  1282. }
  1283. firstLoading.value = false;
  1284. }
  1285. };
  1286. //视频播放结束时触发
  1287. const videoEnd = () => {
  1288. nextTick(() => {
  1289. proxy.refs.video.onended = () => {
  1290. currentDefaultImage.value = 0;
  1291. defaultImage.value = goodsDetail.data.defaultProduct.goodsPics[0];
  1292. };
  1293. });
  1294. };
  1295. // 促销活动信息
  1296. const getPreSell = async (productId) => {
  1297. let param = {
  1298. productId: productId,
  1299. promotionId: goodsDetail.data.defaultProduct.promotionId,
  1300. };
  1301. const { data: value } = await useFetchRaw(
  1302. apiUrl + "v3/promotion/front/preSell/detail",
  1303. { params: param }
  1304. );
  1305. const res = value.value._rawValue;
  1306. if (res.state == 200) {
  1307. let now = new Date();
  1308. let preStartDate = new Date(res.data.startTime);
  1309. let preEndDate = new Date(res.data.endTime);
  1310. preSellInfo.data = res.data;
  1311. let countTime = 0;
  1312. preSellInfo.data.endTime = formatPreTime(new Date(res.data.endTime));
  1313. preSellInfo.data.startTime = formatPreTime(new Date(res.data.startTime));
  1314. if (now > preStartDate && now < preEndDate) {
  1315. preSellInfo.data.pre_run = 2; //活动进行中
  1316. countTime = res.data.distanceEndTime;
  1317. countDown(countTime);
  1318. } else if (now < preStartDate) {
  1319. preSellInfo.data.pre_run = 1; //活动未开始
  1320. countTime =
  1321. (new Date(res.data.startTime).getTime() - now.getTime()) / 1000;
  1322. countDown(countTime);
  1323. } else if (now > preEndDate) {
  1324. preSellInfo.data.pre_run = 3; //活动已结束
  1325. }
  1326. genQrcode();
  1327. } else {
  1328. ElMessage.error(res.msg);
  1329. }
  1330. };
  1331. const getPin = () => {
  1332. let param = {
  1333. productId: goodsDetail.data.defaultProduct.productId,
  1334. promotionId: goodsDetail.data.defaultProduct.promotionId,
  1335. };
  1336. get("v3/promotion/front/spell/detail", param).then((res) => {
  1337. if (res.state == 200) {
  1338. pinInfo.data = res.data;
  1339. let countTime = 0;
  1340. let now = new Date();
  1341. let startTime = new Date(res.data.startTime);
  1342. if (now < startTime) {
  1343. countTime = (startTime.getTime() - now.getTime()) / 1000;
  1344. countDown(countTime);
  1345. } else {
  1346. countTime = res.data.distanceEndTime;
  1347. countDown(countTime);
  1348. }
  1349. genQrcode();
  1350. } else {
  1351. ElMessage.error(res.msg);
  1352. }
  1353. });
  1354. };
  1355. const getSeckill = () => {
  1356. let param = {
  1357. productId: goodsDetail.data.defaultProduct.productId,
  1358. promotionId: goodsDetail.data.defaultProduct.promotionId,
  1359. };
  1360. get("v3/promotion/front/seckill/detail", param).then((res) => {
  1361. if (res.state == 200) {
  1362. seckillInfo.data = res.data;
  1363. let now = new Date();
  1364. let countTime = 0;
  1365. let startTime = new Date(res.data.startTime);
  1366. if (seckillInfo.data.state == 1 || seckillInfo.data.state == 2) {
  1367. countTime = res.data.distanceEndTime;
  1368. countDown(countTime);
  1369. } else {
  1370. countTime = startTime.getTime() - now.getTime();
  1371. }
  1372. genQrcode();
  1373. }
  1374. });
  1375. };
  1376. const countDown = (countTime) => {
  1377. secInt.value = setInterval(() => {
  1378. if (countTime == 0) {
  1379. getInitDataDynamic(calcProductId(route.path));
  1380. clearInterval(secInt.value);
  1381. } else {
  1382. countTime--;
  1383. let day = parseInt(countTime / 60 / 60 / 24);
  1384. let hours = parseInt((countTime / 60 / 60) % 24);
  1385. let minutes = parseInt((countTime / 60) % 60);
  1386. let seconds = parseInt(countTime % 60);
  1387. time.day = day;
  1388. time.hours = hours > 9 ? hours : "0" + hours;
  1389. time.minutes = minutes > 9 ? minutes : "0" + minutes;
  1390. time.seconds = seconds > 9 ? seconds : "0" + seconds;
  1391. }
  1392. }, 1000);
  1393. };
  1394. const formatPreTime = (time) => {
  1395. let op = new Date(time);
  1396. let year = op.getFullYear();
  1397. let month = op.getMonth() + 1;
  1398. let day = op.getDate();
  1399. let hour = op.getHours();
  1400. let minute = op.getMinutes();
  1401. let part1 = [year, month, day]
  1402. .map((i) => (i.toString().length < 2 ? `0${i}` : i))
  1403. .join("-");
  1404. let part2 = [hour, minute]
  1405. .map((i) => (i.toString().length < 2 ? `0${i}` : i))
  1406. .join(":");
  1407. return part1 + " " + part2;
  1408. };
  1409. const getLadder = () => {
  1410. let param = {
  1411. productId: goodsDetail.data.defaultProduct.productId,
  1412. promotionId: goodsDetail.data.defaultProduct.promotionId,
  1413. };
  1414. get("v3/promotion/front/ladder/group/detail", param).then((res) => {
  1415. if (res.state == 200) {
  1416. ladderInfo.data = res.data;
  1417. let now = new Date();
  1418. let countTime = 0;
  1419. let startTime = new Date(res.data.startTime);
  1420. if (now < startTime) {
  1421. countTime = (startTime.getTime() - now.getTime()) / 1000;
  1422. countDown(countTime);
  1423. ladderInfo.data.state = 1;
  1424. } else {
  1425. countTime = res.data.distanceEndTime;
  1426. countDown(countTime);
  1427. ladderInfo.data.state = 2;
  1428. }
  1429. genQrcode();
  1430. }
  1431. });
  1432. };
  1433. const genQrcode = () => {
  1434. if (judgeStock.value) {
  1435. return;
  1436. }
  1437. proxy.$nextTick(() => {
  1438. let canvas = qrcanvas({
  1439. data: goodsDetail.data.shareLink, //二维码内容
  1440. size: 125,
  1441. colorDark: "red",
  1442. });
  1443. setTimeout(() => {
  1444. document.getElementById("qrcodeAct").innerHTML = "";
  1445. document.getElementById("qrcodeAct").appendChild(canvas);
  1446. }, 10);
  1447. });
  1448. };
  1449. // 促销活动信息end
  1450. const recomOffShop = reactive({ data: [] });
  1451. const getRecom = () => {
  1452. get("v3/goods/front/goods/goodsList", {
  1453. storeId: goodsDetail.data.categoryId1,
  1454. }).then((res) => {
  1455. if (res.state == 200) {
  1456. let top = Math.floor(Math.random() * (res.data.list.length - 8)) + 8;
  1457. let end = top - 8;
  1458. recomOffShop.data = res.data.list
  1459. .filter(
  1460. (item) =>
  1461. item.defaultProductId != goodsDetail.data.defaultProduct.productId
  1462. )
  1463. .slice(end, top);
  1464. }
  1465. });
  1466. };
  1467. //添加足迹
  1468. const addLog = () => {
  1469. let params = {
  1470. productId: productId.value,
  1471. };
  1472. post("v3/member/front/productLookLog/add", params).then((res) => {});
  1473. };
  1474. //记录下载
  1475. const downloadAdd = (item) => {
  1476. let url = item.annexUrl;
  1477. let name = item.annexName;
  1478. if (filtersStore.getLoginFlag) {
  1479. post("v3/member/front/download/add", {
  1480. goodsId: item.goodsId,
  1481. annexId: item.annexId,
  1482. }).then((res) => {
  1483. });
  1484. }
  1485. const link = document.createElement("a");
  1486. fetch(url)
  1487. .then((res) => res.blob())
  1488. .then((blob) => {
  1489. link.href = URL.createObjectURL(blob);
  1490. link.download = name;
  1491. document.body.appendChild(link);
  1492. link.click();
  1493. window.URL.revokeObjectURL(link.href);
  1494. document.body.removeChild(link);
  1495. });
  1496. };
  1497. /**
  1498. * 选择规格值
  1499. * @param type:类型 值:choice,规格选择 default:默认
  1500. * @param specId:父级规格值
  1501. * @param specValueId:点击的当前的规格值
  1502. */
  1503. const selectSpecVal = (type, specId, specValueId) => {
  1504. isChoice.value = type == "choice" ? "choice" : "default";
  1505. let curParSpec = []; //当前点击的规格的父级id的当前项
  1506. curParSpec = goodsDetail.data.specs.filter((item) => item.specId == specId);
  1507. let curSPec = []; //当前点击的规格的规格id的当前项
  1508. curSPec = curParSpec[0].specValueList.filter(
  1509. (item1) => item1.specValueId == specValueId
  1510. );
  1511. curSPec[0].checkState = 1;
  1512. //被选择的规格值的id
  1513. let choiceSpecIds = [];
  1514. goodsDetail.data.specs.forEach((item) => {
  1515. if (item.specId != specId) {
  1516. item.specValueList.forEach((item1) => {
  1517. if (item1.checkState == "1") {
  1518. // checkState: 1-选中,2-可选,3-禁用
  1519. choiceSpecIds.push(item1.specValueId);
  1520. }
  1521. });
  1522. } else {
  1523. choiceSpecIds.push(specValueId);
  1524. }
  1525. });
  1526. let params = {
  1527. goodsId: goodsDetail.data.goodsId,
  1528. specValueIds: choiceSpecIds.join(","),
  1529. };
  1530. get("v3/goods/front/goods/productInfo", params).then((res) => {
  1531. if (res.state == 200) {
  1532. let result = res.data;
  1533. goodsDetail.data.defaultProduct = result.defaultProduct;
  1534. productId.value = result.defaultProduct.productId;
  1535. goodsDetail.data.specs = result.specs;
  1536. defaultImage.value = goodsDetail.data.defaultProduct.goodsPics[0];
  1537. currentDefaultImage.value = 0;
  1538. isShowQr.value = false;
  1539. // getInitDataDynamic(productId.value)
  1540. }
  1541. });
  1542. };
  1543. //改变数量按钮样式
  1544. const disStyle = reactive({
  1545. //目的是进入商品详情页面就让减按钮呈现禁止状态
  1546. color: "#DDDDDD",
  1547. backgroundColor: "#F8F8F8",
  1548. });
  1549. watch(
  1550. () => currentSpecNum.value,
  1551. () => {
  1552. //监听数量对加和减的样式做出调整
  1553. let productStock = goodsDetail.data.defaultProduct.productStock;
  1554. if (goodsDetail.data.defaultProduct.productStock == 0) {
  1555. productStock = 999;
  1556. }
  1557. if (currentSpecNum.value >= productStock) {
  1558. proxy.refs.add.style.color = "#DDDDDD";
  1559. proxy.refs.add.style.backgroundColor = "#F8F8F8";
  1560. } else if (currentSpecNum.value <= 1) {
  1561. disStyle.color = "#DDDDDD";
  1562. disStyle.backgroundColor = "#F8F8F8";
  1563. } else {
  1564. disStyle.color = "";
  1565. disStyle.backgroundColor = "";
  1566. proxy.refs.add.style.color = "";
  1567. proxy.refs.add.style.backgroundColor = "";
  1568. }
  1569. }
  1570. );
  1571. watch(currentSpecNum, () => {
  1572. if (currentSpecNum.value > goodsDetail.data.defaultProduct.productStock) {
  1573. currentSpecNum.value = goodsDetail.data.defaultProduct.productStock;
  1574. }
  1575. let reg = /\./g;
  1576. let reg0 = /0+\d/;
  1577. if (
  1578. currentSpecNum.value &&
  1579. (reg.test(currentSpecNum.value) || currentSpecNum.value <= 0)
  1580. ) {
  1581. currentSpecNum.value = 1;
  1582. }
  1583. });
  1584. //发送询盘
  1585. const goBuy = () => {
  1586. enquiryVis.value = true;
  1587. };
  1588. //关闭登录弹框
  1589. const closeLoingModal = () => {
  1590. loginModalVisibleFlag.value = false;
  1591. };
  1592. //关闭登录弹框
  1593. const closeEnquiryModal = () => {
  1594. enquiryVis.value = false;
  1595. };
  1596. //获取看了又看商品(人气数)
  1597. const getRecommend = async () => {
  1598. let params = {
  1599. storeId: vid.value,
  1600. sort: 5,
  1601. pageSize: 3,
  1602. current: 1,
  1603. };
  1604. const { data: value } = await useFetchRaw(
  1605. apiUrl + "v3/goods/front/goods/goodsList",
  1606. { params: params, key: params.sort.toString() }
  1607. );
  1608. const res = value._rawValue;
  1609. if (res.state == 200) {
  1610. let result = res.data;
  1611. recommendeList.data = result.list;
  1612. recommendeList.data.map(
  1613. (item) => (item.goodsPrice = new Number(item.goodsPrice).toFixed(2))
  1614. );
  1615. } else {
  1616. ElMessage.error(res.msg);
  1617. }
  1618. };
  1619. //获取店铺推荐的商品(销量数);获取热门收藏的商品(收藏数) type:recommend销量数 collection:收藏数
  1620. const getStorePopular = async (type) => {
  1621. if (type == "" || !type) {
  1622. type = "recommend";
  1623. } else {
  1624. storePopularType.value = type;
  1625. }
  1626. let params = {
  1627. storeId: vid.value,
  1628. sort:
  1629. storePopularType.value == "recommend"
  1630. ? 1
  1631. : storePopularType.value == "collection"
  1632. ? 6
  1633. : "",
  1634. pageSize: 6,
  1635. current: 1,
  1636. };
  1637. const { data: value } = await useFetchRaw(
  1638. apiUrl + "v3/goods/front/goods/goodsList",
  1639. { params: params, key: params.sort.toString() }
  1640. );
  1641. const res = value._rawValue;
  1642. if (res.state == 200) {
  1643. let result = res.data;
  1644. storePopularList.data = result.list;
  1645. storePopularList.data.map(
  1646. (item) => (item.goodsPrice = new Number(item.goodsPrice).toFixed(2))
  1647. );
  1648. } else {
  1649. ElMessage.error(res.msg);
  1650. }
  1651. };
  1652. //切换商品详情,评价,服务,热销
  1653. const goodsDescType = (type) => {
  1654. // let elementScrollTop =
  1655. // window.pageYOffset ||
  1656. // document.documentElement.scrollTop ||
  1657. // document.body.scrollTop; //获取页面滚动高度
  1658. // if (elementScrollTop > containerTop.value + 33 + 48) {
  1659. // window.scrollTo({
  1660. // top: containerTop.value,
  1661. // behavior: "smooth",
  1662. // });
  1663. // }
  1664. if (type == "" || !type) {
  1665. goodsDesctionType.value = "detail";
  1666. } else {
  1667. goodsDesctionType.value = type;
  1668. }
  1669. if (type == "salestore") {
  1670. getSalestore();
  1671. }
  1672. };
  1673. //获取商品评价
  1674. const getEvaluation = (type) => {
  1675. if (!type) {
  1676. evaluationType.value = "";
  1677. } else {
  1678. evaluationType.value = type;
  1679. }
  1680. let params = {
  1681. productId: productId.value,
  1682. current: evaluationCurrent.value,
  1683. pageSize: evaluationPageSize.value,
  1684. type: evaluationType.value,
  1685. };
  1686. get("v3/goods/front/goods/comment", params).then((res) => {
  1687. if (res.state == 200) {
  1688. let result = res.data;
  1689. goodsCommentsInfo.data = result;
  1690. score.value = Number(goodsCommentsInfo.data.avgScore);
  1691. goodsCommentsInfo.data &&
  1692. goodsCommentsInfo.data.list &&
  1693. goodsCommentsInfo.data.list.length > 0 &&
  1694. goodsCommentsInfo.data.list.map((commentsItem) => {
  1695. commentsItem.memberName =
  1696. commentsItem.memberName.slice(0, 1) +
  1697. "***" +
  1698. commentsItem.memberName.slice(
  1699. commentsItem.memberName.length - 1,
  1700. commentsItem.memberName.length
  1701. );
  1702. });
  1703. } else {
  1704. ElMessage.error(res.msg);
  1705. }
  1706. });
  1707. };
  1708. //评价列表上一页
  1709. const handlePrevCilickChange = () => {
  1710. if (evaluationCurrent.value == 1) {
  1711. evaluationCurrent.value = 1;
  1712. } else {
  1713. evaluationCurrent.value--;
  1714. }
  1715. getEvaluation(evaluationType.value);
  1716. };
  1717. //评价列表下一页
  1718. const handleNextCilickChange = () => {
  1719. if (evaluationCurrent.value >= goodsCommentsInfo.data.list.length) {
  1720. evaluationCurrent.value = goodsCommentsInfo.data.list.length;
  1721. } else {
  1722. evaluationCurrent.value++;
  1723. }
  1724. getEvaluation(evaluationType.value);
  1725. };
  1726. //评价列表当前选择页
  1727. const handleCurrentChange = (val) => {
  1728. evaluationCurrent.value = val;
  1729. getEvaluation(evaluationType.value);
  1730. };
  1731. //获取热销店铺推荐
  1732. const getSalestore = () => {
  1733. let params = {
  1734. storeId: goodsDetail.data.storeInf.storeId,
  1735. sort: 7,
  1736. pageSize: salesPageSize.value,
  1737. current: salesCurrent.value,
  1738. };
  1739. get("v3/goods/front/goods/goodsList", params).then((res) => {
  1740. if (res.state == 200) {
  1741. let result = res.data;
  1742. recommendeData.data = result;
  1743. recommendedList.data = result.list;
  1744. recommendedList.data.map(
  1745. (item) => (item.goodsPrice = new Number(item.goodsPrice).toFixed(2))
  1746. );
  1747. } else {
  1748. ElMessage.error(res.msg);
  1749. }
  1750. });
  1751. };
  1752. //店铺热销列表上一页
  1753. const handlePrevCilickChangeSales = () => {
  1754. if (evaluationCurrent.value == 1) {
  1755. salesCurrent.value = 1;
  1756. } else {
  1757. salesCurrent.value--;
  1758. }
  1759. getSalestore();
  1760. };
  1761. //店铺热销列表下一页
  1762. const handleNextCilickChangeSales = () => {
  1763. if (salesCurrent.value >= goodsCommentsInfo.data.list.length) {
  1764. salesCurrent.value = goodsCommentsInfo.data.list.length;
  1765. } else {
  1766. salesCurrent.value++;
  1767. }
  1768. getSalestore();
  1769. };
  1770. //店铺热销列表当前选择页
  1771. const handleCurrentChangeSales = (val) => {
  1772. salesCurrent.value = val;
  1773. getSalestore();
  1774. };
  1775. //关注店铺及取消关注
  1776. const headercat = ref(null)
  1777. const focusStore = () => {
  1778. if (filtersStore.getLoginFlag) {
  1779. //已登录
  1780. let params = {
  1781. storeIds: goodsDetail.data.storeInf.storeId,
  1782. isCollect: !goodsDetail.data.storeInf.isFollowStore,
  1783. };
  1784. post("v3/member/front/followStore/edit", params).then((res) => {
  1785. if (res.state == 200) {
  1786. goodsDetail.data.storeInf.isFollowStore =!goodsDetail.data.storeInf.isFollowStore;
  1787. proxy.refs.headercat.setfollowStore(goodsDetail.data.storeInf.isFollowStore? "true" : "false")
  1788. if (goodsDetail.data.storeInf.isFollowStore) {
  1789. sldStatEvent({
  1790. behaviorType: "fol",
  1791. storeId: goodsDetail.data.storeInf.storeId,
  1792. });
  1793. }
  1794. }
  1795. });
  1796. } else {
  1797. //未登录提示登录
  1798. loginModalVisibleFlag.value = true;
  1799. }
  1800. };
  1801. //商品收藏及取消收藏
  1802. const collectGoods = () => {
  1803. if (filtersStore.getLoginFlag) {
  1804. //已登录
  1805. let params = {
  1806. productIds: productId.value,
  1807. isCollect: !goodsDetail.data.followGoods,
  1808. };
  1809. post("v3/member/front/followProduct/edit", params).then((res) => {
  1810. if (res.state == 200) {
  1811. goodsDetail.data.followGoods = !goodsDetail.data.followGoods;
  1812. if (goodsDetail.data.followGoods) {
  1813. sldStatEvent({
  1814. behaviorType: "fav",
  1815. goodsId: goodsDetail.data.goodsId,
  1816. storeId: goodsDetail.data.storeInf.storeId,
  1817. });
  1818. }
  1819. } else {
  1820. ElMessage.error(res.msg);
  1821. }
  1822. });
  1823. } else {
  1824. //未登录提示登录
  1825. loginModalVisibleFlag.value = true;
  1826. }
  1827. };
  1828. //点击查看全部查看全部的商品规格参数
  1829. const openParameter = () => {
  1830. parameterShow.value = !parameterShow.value;
  1831. };
  1832. //选择商品主图
  1833. const selectDefaultImage = (goodsImgItem, goodsImgIndex) => {
  1834. defaultImage.value = goodsImgItem;
  1835. currentDefaultImage.value = goodsImgIndex;
  1836. };
  1837. //切换商品主图
  1838. const switchDefaultImage = (type) => {
  1839. let defaultImagelength = goodsDetail.data.defaultProduct.goodsPics.length;
  1840. if (type == "left") {
  1841. currentDefaultImage.value--;
  1842. if (currentDefaultImage.value <= 0) {
  1843. currentDefaultImage.value = 0;
  1844. }
  1845. defaultImage.value =
  1846. goodsDetail.data.defaultProduct.goodsPics[currentDefaultImage.value];
  1847. } else {
  1848. currentDefaultImage.value++;
  1849. if (currentDefaultImage.value >= defaultImagelength) {
  1850. currentDefaultImage.value = defaultImagelength - 1;
  1851. }
  1852. defaultImage.value =
  1853. goodsDetail.data.defaultProduct.goodsPics[currentDefaultImage.value];
  1854. }
  1855. goodsPictureListsLeft();
  1856. };
  1857. //图片列表的left移动的距离
  1858. const goodsPictureListsLeft = () => {
  1859. //获取 goods_picture_list 的元素
  1860. let goodsPictureLists = goodsPictureList.value;
  1861. //列表默认显示5张图片
  1862. if (
  1863. goodsDetail.data.defaultProduct.goodsPics.length > 5 &&
  1864. currentDefaultImage.value >= 0
  1865. ) {
  1866. /* 分析找规律:
  1867. 如果有8张图片,点击右键 最大可以向左移动的距离为 8 - 5既3张图,left移动的距离为 3 * - 66px; currentDefaultImage.value从0开始的
  1868. 其中:66px为每一个元素需要每次移动的距离 为图片的宽度与图片之间的间距的和
  1869. currentDefaultImage.value == 4;goodsPictureLists.style.left = 0;
  1870. currentDefaultImage.value == 5;goodsPictureLists.style.left = (5-4) (1) * -66px;
  1871. currentDefaultImage.value == 6;goodsPictureLists.style.left = (5-2) (2) * -66px;
  1872. currentDefaultImage.value == 7;goodsPictureLists.style.left = (5-3) (2) * -66px;
  1873. 点击左键,最大可以向右移动的距离为 8 - 5 即3张
  1874. currentDefaultImage.value == 0;goodsPictureLists.style.left = 0;
  1875. currentDefaultImage.value == 1; goodsPictureLists.style.left = -66px;
  1876. currentDefaultImage.value == 2; goodsPictureLists.style.left = 2 * -66px;
  1877. currentDefaultImage.value == 3; goodsPictureLists.style.left = 3 * -66px;
  1878. */
  1879. if (currentDefaultImage.value > 4) {
  1880. goodsPictureLists.style.left =
  1881. (currentDefaultImage.value - 4) * -66 + "px";
  1882. }
  1883. if (
  1884. currentDefaultImage.value <
  1885. goodsDetail.data.defaultProduct.goodsPics.length - 4
  1886. ) {
  1887. goodsPictureLists.style.left = currentDefaultImage.value * -66 + "px";
  1888. }
  1889. }
  1890. };
  1891. //获取元素距离父元素的顶部及左边的距离
  1892. const offset = (el) => {
  1893. if (process.client) {
  1894. let top = el.offsetTop;
  1895. let left = el.offsetLeft;
  1896. if (el.offsetParent) {
  1897. el = el.offsetParent;
  1898. top += el.offsetTop;
  1899. left += el.offsetLeft;
  1900. }
  1901. return {
  1902. left: left,
  1903. top: top,
  1904. };
  1905. }
  1906. };
  1907. if (process.client) {
  1908. nextTick(() => {
  1909. containerTop.value = offset(container.value).top;
  1910. window.addEventListener("scroll", scrollHandle); //绑定页面滚动事件
  1911. window.addEventListener("click", () => {
  1912. addrDialogVisible.value = false;
  1913. });
  1914. });
  1915. }
  1916. //鼠标移动
  1917. const move = (e) => {
  1918. if (currentDefaultImage.value == -1) {
  1919. return;
  1920. }
  1921. //主图父元素的信息 宽,高
  1922. let previewsBox = previewBox.value;
  1923. let previewBoxWidth = previewsBox.offsetWidth;
  1924. let previewBoxHeight = previewsBox.offsetHeight;
  1925. //主图父元素距离顶部的距离
  1926. let previewsBoxLeft = offset(previewsBox).left;
  1927. let previewsBoxTop = offset(previewsBox).top;
  1928. // 遮罩盒子的信息宽,高
  1929. let masksBox = maskBox.value;
  1930. let maskBoxWidth = masksBox.offsetWidth;
  1931. let maskBoxHeight = masksBox.offsetHeight;
  1932. //鼠标距离屏幕距离
  1933. let moveX = e.clientX;
  1934. let moveY = e.clientY;
  1935. //获取左侧大图父元素的信息
  1936. let zoomsBox = zoomBox.value;
  1937. let zoomBoxWidth = zoomsBox.offsetWidth;
  1938. let zoomBoxHeight = zoomsBox.offsetHeight;
  1939. // 获取大图元素的信息宽,高
  1940. let pictureBigBox = pictureBig.value;
  1941. let pictureBigWidth = pictureBigBox.offsetWidth;
  1942. let pictureBigHeight = pictureBigBox.offsetHeight;
  1943. //获取滚动条的高度
  1944. let scroll = document.documentElement.scrollTop || document.body.scrollTop;
  1945. //主图距离父元素的left及top值
  1946. let left = moveX - previewsBoxLeft - maskBoxWidth / 2;
  1947. let top;
  1948. if (scroll > 0) {
  1949. top = moveY - previewsBoxTop + scroll - maskBoxHeight / 2;
  1950. } else {
  1951. top = moveY - previewsBoxTop - maskBoxHeight / 2;
  1952. }
  1953. //移动限制最大宽度,及最大高度
  1954. let maxWidth = previewBoxWidth - maskBoxWidth;
  1955. let maxHeight = previewBoxHeight - maskBoxHeight;
  1956. left = left < 0 ? 0 : left > maxWidth ? maxWidth : left;
  1957. top = top < 0 ? 0 : top > maxHeight ? maxHeight : top;
  1958. //比列
  1959. let parcentX = left / maxWidth;
  1960. let parcentY = top / maxHeight;
  1961. //遮罩层的定位值
  1962. maskBox.value.style.left = left + "px";
  1963. maskBox.value.style.top = top + "px";
  1964. //大图元素的定位值
  1965. pictureBig.value.style.left =
  1966. parcentX * (zoomBoxWidth - pictureBigWidth) + "px";
  1967. pictureBig.value.style.top =
  1968. parcentY * (zoomBoxHeight - pictureBigHeight) + "px";
  1969. pictureBig.value.style.width =
  1970. (previewBoxWidth / maskBoxWidth) * zoomBoxWidth + "px";
  1971. pictureBig.value.style.height =
  1972. (previewBoxHeight / maskBoxHeight) * zoomBoxHeight + "px";
  1973. };
  1974. //鼠标移出
  1975. const out = () => {
  1976. maskShow.value = false;
  1977. };
  1978. //鼠标移入
  1979. const enter = () => {
  1980. if (currentDefaultImage.value == -1) {
  1981. return;
  1982. }
  1983. addrDialogVisible.value = false;
  1984. maskShow.value = true;
  1985. };
  1986. // 评论区查看图片
  1987. const showImg = (index, img) => {
  1988. imgSource.value = img;
  1989. imgIndex.value = index;
  1990. imgVisible.value = true;
  1991. };
  1992. //分享
  1993. const share = (type) => {
  1994. let title = goodsDetail.data.goodsName; //需要分享的标题,这里取商品名字
  1995. let url = goodsDetail.data.shareLink; //分享的地址,用户点击可以进入到该商品
  1996. let content = goodsDetail.data.goodsBrief; //自定义内容,这里取商品广告词
  1997. let targetUrl = ""; //跳转的url地址
  1998. if (type == "weixin") {
  1999. wxShareCode.value = true;
  2000. //微信
  2001. let canvas = qrcanvas({
  2002. data: url, //二维码内容
  2003. size: 100,
  2004. colorDark: "red",
  2005. });
  2006. let share_wx_qrcode = document.getElementById("share_wx_qrcode");
  2007. if (
  2008. wxShareCode.value &&
  2009. share_wx_qrcode != null &&
  2010. share_wx_qrcode != undefined
  2011. ) {
  2012. document.getElementById("share_wx_qrcode").innerHTML = "";
  2013. document.getElementById("share_wx_qrcode").appendChild(canvas);
  2014. }
  2015. } else if (type == "qzone") {
  2016. wxShareCode.value = false;
  2017. //QQ空间
  2018. targetUrl =
  2019. "https://sns.qzone.qq.com/cgi-bin/qzshare/cgi_qzshare_onekey?title=" +
  2020. encodeURIComponent(title) +
  2021. "&desc=" +
  2022. encodeURIComponent(content) +
  2023. "&summary=" +
  2024. encodeURIComponent(content) +
  2025. "&url=" +
  2026. encodeURIComponent(url);
  2027. window.open(targetUrl, "_blank");
  2028. } else if (type == "sina") {
  2029. wxShareCode.value = false;
  2030. //新浪微博
  2031. targetUrl =
  2032. "https://service.weibo.com/share/share.php?title=" +
  2033. encodeURIComponent(content + "「" + title + "」" + " 点这里" + url);
  2034. window.open(targetUrl, "_blank");
  2035. }
  2036. };
  2037. const refreshInfo = () => {
  2038. history.go(0);
  2039. };
  2040. const updateFllow = (e) => {
  2041. goodsDetail.data.storeInf.isFollowStore = e.state == "true" ? true : false;
  2042. };
  2043. //暴露的变量及方法
  2044. </script>
  2045. <style lang="scss">
  2046. @import "@/assets/style/base.scss";
  2047. @import "@/assets/style/theme.scss";
  2048. @import "@/assets/style/iconfont.css";
  2049. @import "@/assets/style/goodsDetail.scss";
  2050. .popular_list_empty {
  2051. height: 95px;
  2052. font-size: 14px;
  2053. /*font-family: Microsoft YaHei;*/
  2054. font-weight: 400;
  2055. color: #666666;
  2056. }
  2057. .imageBorder {
  2058. border: 1px solid #eee;
  2059. }
  2060. .goods_picture_big {
  2061. border: 1px solid #eee;
  2062. }
  2063. .el-radio__inner:hover {
  2064. border-color: $colorMain;
  2065. }
  2066. .el-radio__input.is-checked .el-radio__inner {
  2067. border-color: $colorMain;
  2068. background: $colorMain;
  2069. }
  2070. .el-radio__input.is-checked + .el-radio__label {
  2071. color: $colorMain;
  2072. }
  2073. .el-radio {
  2074. margin-bottom: 10px;
  2075. display: flex;
  2076. align-items: flex-start;
  2077. white-space: unset;
  2078. margin-right: unset;
  2079. }
  2080. .el-radio-button__inner,
  2081. .el-radio-group {
  2082. /* display: block; */
  2083. line-height: 1;
  2084. vertical-align: middle;
  2085. }
  2086. .el-radio__label {
  2087. font-size: 13px;
  2088. width: 320px;
  2089. overflow: hidden;
  2090. text-overflow: ellipsis;
  2091. display: -webkit-box;
  2092. -webkit-line-clamp: 2;
  2093. -webkit-box-orient: vertical;
  2094. word-break: break-all;
  2095. line-height: 22px;
  2096. margin-top: -5px;
  2097. }
  2098. .evaluationes {
  2099. color: #3b4 !important;
  2100. }
  2101. .sld_goods_detail .goods_htmls .ql-video {
  2102. width: 525px;
  2103. height: 315px;
  2104. }
  2105. .sld_goods_detail .goods_htmls a {
  2106. display: inline-block;
  2107. margin: 5px auto;
  2108. color: #0000ff;
  2109. text-decoration: underline;
  2110. }
  2111. .sld_goods_detail .goods_htmls table {
  2112. border-collapse: collapse;
  2113. padding: 0;
  2114. }
  2115. .sld_goods_detail .goods_htmls td,
  2116. .sld_goods_detail .goods_htmls th {
  2117. border: 1px solid #ddd;
  2118. padding: 5px 10px;
  2119. }
  2120. .sld_goods_detail .goods_htmls ol li,
  2121. .sld_goods_detail .goods_htmls ul li {
  2122. list-style: unset;
  2123. }
  2124. .sld_goods_detail {
  2125. .summary {
  2126. .coupon {
  2127. .el-dialog__header {
  2128. padding-top: 18px;
  2129. padding-bottom: 18px;
  2130. .el-dialog__title {
  2131. color: #333333;
  2132. /*font-family: Microsoft YaHei;*/
  2133. font-weight: bold;
  2134. }
  2135. .el-dialog__close {
  2136. color: #333333;
  2137. font-size: 20px;
  2138. }
  2139. }
  2140. .el-dialog__body {
  2141. background: #f8f8f8;
  2142. }
  2143. }
  2144. }
  2145. }
  2146. .summary-info{
  2147. }
  2148. .mt-20{
  2149. margin-top: 20px;
  2150. }
  2151. .summary_htmls img{
  2152. width: 100%;
  2153. }
  2154. </style>